home *** CD-ROM | disk | FTP | other *** search
- /***************************************************************************
- * *
- * File Name: GRABIMG.C *
- * Description: Build Image (IM) file from on-screen image. *
- * Notes: This program displays a PCX picture and allows you to *
- * click inside a boundary of colour 255. It then grabs *
- * everthing within that boundary and saves it as an *
- * image. You can keep clicking if you want a "bank" of *
- * two or more images. *
- * *
- * Special Note: The mouse pointer in this program is an excellent *
- * example of how you can use the image system as a *
- * sprite system. The "proper" version of this program *
- * uses the full-blown sprite system to display the *
- * mouse, so I rewrote it to show you how you can use *
- * just the image system described this month to do some *
- * quite nice stuff. *
- * *
- * As of this month, there'll be three files included on *
- * the disk:- *
- * HIGHC.OBJ All the functions we've developed so far *
- * HIGHC.C The source to them *
- * HIGHC.H A header file for structures and soforth *
- * *
- * You MUST tell your compiler/linker to include *
- * HIGHC.OBJ at link time, otherwise it won't be able to *
- * find all the "basic" functions like g_SetVGA. *
- * *
- ***************************************************************************/
-
- #include "highc.h" // From now on, include this in every program
- #include "alloc.h"
- #include "stdlib.h"
- #include "stdio.h"
- #include "string.h"
-
- #define NO_FILE -1
- #define NOT_KNOWN -2
- #define PCX 1 // Just to make it read a bit nicer
-
- typedef struct {
- char far *ptr;
- int h, w;
- int t, l;
- char far *sptr;
- } IMAGE_DEFN;
-
- IMAGE_DEFN img[30];
-
- char *gen_IM(char *string);
-
- main(int argc, char *argv[])
- {
-
- char *IM_file;
- char far *t;
- signed char x;
- unsigned char w, h, contin;
- char signature[25];
- FILE *fp;
-
-
- if (argc != 3)
- {
- printf("Usage: GRABIMG {PCX_Picture_file} {Image_file}\n");
- exit(0);
- }
-
- sbptr = farcalloc(64000, 1); // Allocate the screen buffer
- palette = farcalloc(768, 1); // Allocate the palette space
-
- g_SetVGA();
- x=g_LoadScreen(argv[1]);
-
- if (x != PCX)
- {
- g_SetTxt();
- printf("Error! Non-existent or non-PCX file %s\n", argv[1]);
- exit(0);
- }
-
- IM_file = gen_IM(argv[2]); // If the user hasn't entered the ".IM"
- // extension, add it.
-
- fp = fopen(IM_file, "rb");
-
- if (fp == NULL)
- {
- fp = fopen(IM_file, "wb");
- fwrite("Image File", 1, 10, fp);
- }
- else
- {
- fseek(fp, 0L, SEEK_SET);
- fread(signature, 1, 10, fp);
- if (strncmp(signature, "Image File", 10) != 0)
- {
- g_SetTxt();
- printf("Error! File %s is not a valid Image file\n", IM_file);
- exit(0);
- }
- fclose(fp);
- fp = fopen(IM_file, "ab");
- }
-
- load_img("grabimg.im", 0); // The image bank that contains my mouse pointer
- pasteimg(0, 160, 100); // Bung it out on screen
-
- contin = 1;
-
- while (contin)
- {
- MCheck();
- if (m.lmr)
- add_im(fp);
- if (m.rmr)
- contin = 0;
- update_mouse(); // Remove the pointer from screen & redraw it
- g_SwapScr();
- }
-
- fclose(fp);
-
- g_SetTxt();
-
- return(OK);
- }
-
-
- char *gen_PCX(char *filnam)
- {
-
- static char pcxnam[100];
- char *s1, ss;
-
- if ( (s1 = strstr(strupr(filnam), ".PCC")) == NULL)
- {
- if ( (s1 = strstr(strupr(filnam), ".PCX")) == NULL)
- {
- strcpy(pcxnam, filnam);
- strcat(pcxnam, ".PCX");
- }
- else
- {
- ss = s1 - filnam;
- strncpy(pcxnam, filnam, ss);
- }
- }
- else
- {
- ss = s1 - filnam;
- strncpy(pcxnam, filnam, ss);
- }
-
- s1 = pcxnam;
-
- return(s1);
-
- }
-
-
- char *gen_IM(char *filnam)
- {
-
- static char imnam[100];
- char *s1, ss;
-
- if ( (s1 = strstr(strupr(imnam), ".IM")) == NULL)
- {
- strcpy(imnam, filnam);
- strcat(imnam, ".IM");
- }
- else
- {
- ss = s1 - imnam;
- strncpy(imnam, filnam, ss);
- }
-
-
- return(imnam);
-
- }
-
- add_im(FILE *fp)
- {
-
- int x, y;
- unsigned char ctr;
- int lc, tc, bc, rc;
- char *sptr;
-
- removeimg(0); // Take the pointer off screen - you don't want to save it
- // as part of this image.
-
- /* OK, first things first. Go up until you find a colour 255 pixel. Go back
- down one. Then travel left until you hit a colour 255 pixel. Go one back
- to the right. And there you are, at the top left corner of your box! */
-
- ctr = 0; tc = 0; lc = 0;
- while (ctr++ < 170)
- {
- sptr = g_FPtr(sbptr, ((m.my - ctr) * 320) + m.mx);
- if (*sptr == 255)
- {
- tc = (m.my - ctr) + 1;
- ctr = 170;
- }
- }
-
- if (tc == 0)
- {
- tc = m.my;
- lc = m.mx;
- }
- else
- {
- ctr = 0;
- while (ctr++ < 170)
- {
- sptr = g_FPtr(sbptr, ((m.my * 320) + m.mx) - ctr);
- if (*sptr == 255)
- {
- lc = (m.mx - ctr) + 1;
- ctr = 170;
- }
- }
- }
-
- if (lc == 0)
- lc = m.mx;
-
-
- /* Now find the bottom-right of the box... */
-
- ctr = 0; bc = 0; rc = 0;
- while (ctr++ < 170)
- {
- sptr = g_FPtr(sbptr, ((m.my + ctr) * 320) + m.mx);
- if (*sptr == 255)
- {
- bc = (m.my + ctr) - 1;
- ctr = 170;
- }
- }
-
- if (bc == 0)
- {
- bc = m.my+10;
- rc = m.mx+10;
- }
- else
- {
- ctr = 0;
- while (ctr++ < 170)
- {
- sptr = g_FPtr(sbptr, ((m.my * 320) + m.mx) + ctr);
- if (*sptr == 255)
- {
- rc = (m.mx + ctr) - 1;
- ctr = 170;
- }
- }
- }
-
- if (rc == 0)
- rc = m.mx+10;
-
- fputc(255, fp); // Start of new image marker
- fputc( (char)rc-lc, fp); // Width (0-255)
- fputc( (char)(bc-tc)+1, fp); // Height (0-200)
-
- for (x=0; x <= bc-tc; x++)
- {
- sptr = g_FPtr(sbptr, ((tc + x) * 320) + (lc));
- for (y=0; y < rc-lc; y++)
- fputc(*sptr++, fp);
- }
-
- return(1);
- }
-
-
- update_mouse()
- {
- removeimg(0);
- pasteimg(0, m.mx, m.my);
- return(0);
- }
-
- pasteimg(char imgnum, int l, int t)
-
- {
-
- unsigned int j, k, h, w, m, n;
- char far *sp1;
- char far *sp2;
-
- if (imgnum < 0 || imgnum > 30) // We're only allowing 30 images at one time
- return(-1); // (could easily be increased)
-
- if (l < 0 || t < 0 || l > 319 || t > 199)
- return(-2); // Don't paste an image off the edge of
- // the screen.
-
- if ( (sp2 = img[imgnum].ptr) == NULL)
- return(-3); // The requested image has no image data
-
- w = img[imgnum].w;
- h = img[imgnum].h;
-
- if (img[imgnum].sptr != NULL) // If, for some reason, the background is
- farfree(img[imgnum].sptr); // already saved, throw it away
-
- if ( (img[imgnum].sptr = farcalloc(w*h, 1)) == NULL)
- return(-4); // Not enough memory to save background
-
- save_IMG_back(imgnum, l, t);
-
- m = 0;
-
- for (j = 0; j < h; j++)
- {
- if (j+t < 200)
- {
- sp1 = g_FPtr(sbptr, (t * 320) + (j * 320) + l);
- n = 0;
- for (k = 0; k < w; k++)
- {
- if (sp2[m] != 0 && k+l < 320)
- sp1[n] = sp2[m];
- n++; m++;
- }
- }
- }
-
- img[imgnum].t = t;
- img[imgnum].l = l;
-
- return(OK);
-
- }
-
-
- save_IMG_back(imgnum, l, t)
-
- int l, t;
- char imgnum;
-
- {
-
- unsigned int j, k, w, h, n, m;
- char far *sp1;
- char far *sp2;
-
- sp2 = img[imgnum].sptr;
- w = img[imgnum].w;
- h = img[imgnum].h;
- m = 0;
-
- for (j = 0; j < h; j++)
- {
- sp1 = g_FPtr(sbptr, (t * 320) + (j * 320) + l);
- n = 0;
- for (k = 0; k < w; k++)
- sp2[m++] = sp1[n++];
- }
-
- return(OK);
-
- }
-
-
- removeimg(imgnum)
- char imgnum;
-
- {
-
- unsigned int j, k, n, t, l, w, h, m;
- char far *sp1;
- char far *sp2;
-
- if (img[imgnum].sptr == NULL)
- return(-1); // Image is not on screen
-
- sp2 = img[imgnum].sptr;
- t = img[imgnum].t;
- l = img[imgnum].l;
- w = img[imgnum].w;
- h = img[imgnum].h;
-
- m = 0;
-
-
- for (j = 0; j < h; j++)
- {
- if (j+t < 200)
- {
- sp1 = g_FPtr(sbptr, (t * 320) + (j * 320) + l);
- n = 0;
- for (k = 0; k < w; k++)
- {
- if (k+l < 320)
- sp1[n] = sp2[m];
- n++; m++;
- }
- }
- }
-
- farfree(img[imgnum].sptr);
- img[imgnum].sptr = 0;
-
- return(OK);
-
- }
-
-
- load_img(char *filnam, char append)
- {
-
- int j, k, w, h;
- FILE *fp;
- char far *t;
- char sig[21];
-
- if ( (fp = fopen(filnam, "rb")) == NULL)
- return(NO_FILE);
-
- if (append)
- {
- for (j=0; j<50; j++)
- if (img[j].ptr == NULL)
- break;
- }
- else
- j=0;
-
- fread(sig, 1, 10, fp);
- sig[10] = 0;
-
- if ( strcmp("Image File", sig) != NULL)
- return(NOT_KNOWN);
-
- while (!feof(fp) && j < 50)
- {
- if (fgetc(fp) != 255)
- break;
-
- w = (int)fgetc(fp);
- h = (int)fgetc(fp);
-
- t = farcalloc(w*h, 1);
-
- fread(t, 1, w*h, fp);
-
- img[j].ptr = t;
- img[j].w = w;
- img[j++].h = h;
- }
-
- while (j++ < 50)
- {
- img[j].ptr = NULL;
- img[j].w = 0;
- img[j].h = 0;
- }
-
-
-
- return(OK);
-
- }